import time
from datetime import datetime,timedelta

from flask import render_template, request, current_app, session, redirect, g, jsonify

from info import user_login_data
from info.models import User, News
from info.utils.response_code import RET
from . import admin_blue

# 功能描述: 新闻审核详情
# 请求路径: /admin/news_review_detail
# 请求方式: GET,POST
# 请求参数: GET, news_id, POST,news_id, action
# 返回值:GET,渲染news_review_detail.html页面,data字典数据
@admin_blue.route('/news_review_detail',methods=["GET","POST"])
def news_review_detail():
    """
    - 1.判断请求方式,如果GET,渲染页面
    - 1.1 获取参数,新闻编号
    - 1.2 校验参数
    - 1.3 根据新闻编号获取新闻对象,并判断是否存在
    - 1.4 携带新闻数据,渲染页面
    - 2.如果是POST请求,获取参数
    - 3.校验参数,为空校验
    - 4.操作类型校验
    - 5.通过编号获取新闻对象
    - 6.判断新闻对象是否存在
    - 7.根据操作类型,改变新闻状态
    - 8.返回响应

    :return:
    """
    # - 1.判断请求方式,如果GET,渲染页面
    if request.method == "GET":
        # - 1.1 获取参数,新闻编号
        news_id = request.args.get("news_id")

        # - 1.2 校验参数
        if not news_id: return jsonify(errno=RET.PARAMERR,errmsg="参数不全")

        # - 1.3 根据新闻编号获取新闻对象,并判断是否存在
        try:
            news = News.query.get(news_id)
        except Exception as e:
            current_app.logger.error(e)
            return jsonify(errno=RET.DBERR,errmsg="新闻获取失败")

        if not news: return jsonify(errno=RET.NODATA,errmsg="新闻不存在")

        # - 1.4 携带新闻数据,渲染页面
        return render_template("admin/news_review_detail.html",news=news.to_dict())

    # - 2.如果是POST请求,获取参数
    news_id = request.json.get("news_id")
    action = request.json.get("action")

    # - 3.校验参数,为空校验
    if not all([news_id,action]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不全")

    # - 4.操作类型校验
    if not action in ["accept","reject"]:
        return jsonify(errno=RET.DATAERR,errmsg="操作类型有误")

    # - 5.通过编号获取新闻对象
    try:
        news = News.query.get(news_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="新闻获取失败")

    # - 6.判断新闻对象是否存在
    if not news: return jsonify(errno=RET.NODATA,errmsg="新闻不存在")

    # - 7.根据操作类型,改变新闻状态
    if action == "accept":
        news.status = 0
    else:
        reason = request.json.get("reason","")
        news.reason = reason
        news.status = -1

    # - 8.返回响应
    return jsonify(errno=RET.OK,errmsg="操作成功")


# 功能描述: 新闻审核列表
# 请求路径: /admin/news_review
# 请求方式: GET
# 请求参数: GET, p,keywords
# 返回值:渲染user_list.html页面,data字典数据
@admin_blue.route('/news_review')
def news_review():
    """
    - 1.获取参数
    - 2.参数类型转换
    - 3.分页查询
    - 4.获取分页对象属性,总页数,当前页,当前页对象列表
    - 5.对象列表转成,字典列表
    - 6.拼接数据,渲染页面
    :return:
    """
    # - 1.获取参数
    page = request.args.get("p","1")
    keywords = request.args.get("keywords")

    # - 2.参数类型转换
    try:
        page  = int(page)
    except Exception as e:
        current_app.logger.error(e)
        page = 1

    # - 3.分页查询,未通过和待审核的
    try:

        #判断是否填写了搜索关键字
        filters = [News.status != 0]
        if keywords:
            filters.append(News.title.contains(keywords))

        paginate = News.query.filter(*filters).order_by(News.create_time.desc()).paginate(page,10,False)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="获取用户失败")

    # - 4.获取分页对象属性,总页数,当前页,当前页对象列表
    totalPage = paginate.pages
    currentPage = paginate.page
    items = paginate.items

    # - 5.对象列表转成,字典列表
    news_list = []
    for news in items:
        news_list.append(news.to_review_dict())

    # - 6.拼接数据,渲染页面
    data = {
        "totalPage":totalPage,
        "currentPage":currentPage,
        "news_list":news_list
    }
    return render_template("admin/news_review.html",data=data)

#功能描述: 用户列表
# 请求路径: /admin/user_list
# 请求方式: GET
# 请求参数: p
# 返回值:渲染user_list.html页面,data字典数据
@admin_blue.route('/user_list')
def user_list():
    """
    - 1.获取参数
    - 2.参数类型转换
    - 3.分页查询
    - 4.获取分页对象属性,总页数,当前页,当前页对象列表
    - 5.对象列表转成,字典列表
    - 6.拼接数据,渲染页面
    :return:
    """
    # - 1.获取参数
    page = request.args.get("p","1")

    # - 2.参数类型转换
    try:
        page  = int(page)
    except Exception as e:
        current_app.logger.error(e)
        page = 1

    # - 3.分页查询
    try:
        paginate = User.query.filter(User.is_admin == False).order_by(User.create_time.desc()).paginate(page,10,False)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="获取用户失败")

    # - 4.获取分页对象属性,总页数,当前页,当前页对象列表
    totalPage = paginate.pages
    currentPage = paginate.page
    items = paginate.items

    # - 5.对象列表转成,字典列表
    user_list = []
    for user in items:
        user_list.append(user.to_admin_dict())

    # - 6.拼接数据,渲染页面
    data = {
        "totalPage":totalPage,
        "currentPage":currentPage,
        "user_list":user_list
    }
    return render_template("admin/user_list.html",data=data)

# 功能描述: 管理员用户统计
# 请求路径: /admin/user_count
# 请求方式: GET
# 请求参数: 无
# 返回值:渲染页面user_count.html,字典数据
@admin_blue.route('/user_count')
def user_count():
    """
    - 1.查询总人数,不包含管理员
    - 2.查询月活人数
    - 3.查询日活人数
    - 4.查询时间段内,活跃人数
    - 5.携带数据,渲染页面
    :return:
    """
    # - 1.查询总人数,不包含管理员
    try:
        total_count = User.query.filter(User.is_admin == False).count()
    except Exception as e:
        current_app.logger.error(e)

    # - 2.查询月活人数,获取日历对象
    calender = time.localtime()
    try:
        #2.1 获取到本月的1号: 零点钟
        month_starttime_str = "%d-%d-01"%(calender.tm_year,calender.tm_mon) #2018-10-01
        # strptime(date_str,format) : string parse time参数1:表示需要转换时间字符串   #参数2: 转换成日期的格式
        month_starttime_date = datetime.strptime(month_starttime_str,"%Y-%m-%d")

        #2.2 获取此时的时间
        month_endtime_date = datetime.now()

        #2.3 根据时间,查询用户
        month_count = User.query.filter(User.last_login >= month_starttime_date, User.last_login <= month_endtime_date,User.is_admin==False).count()

    except Exception as e:
        current_app.logger.error(e)

    # - 3.查询日活人数
    try:
        #2.1 获取到本日: 零点钟
        day_starttime_str = "%d-%d-%d"%(calender.tm_year,calender.tm_mon,calender.tm_mday) #2018-10-21
        day_starttime_date = datetime.strptime(day_starttime_str,"%Y-%m-%d")

        #2.2 获取此时的时间
        day_endtime_date = datetime.now()

        #2.3 根据时间,查询用户
        day_count = User.query.filter(User.last_login >= day_starttime_date, User.last_login <= day_endtime_date,User.is_admin==False).count()

    except Exception as e:
        current_app.logger.error(e)

    # - 4.查询时间段内,活跃人数
    active_date = [] #记录活跃日期
    active_count = [] #记录活跃人数
    for i in range(0, 31):
        # 当天开始时间A
        begin_date = day_starttime_date - timedelta(days=i)
        # 当天开始时间, 的后一天B
        end_date = day_starttime_date - timedelta(days=i - 1)

        # 添加当天开始时间字符串到, 活跃日期中
        active_date.append(begin_date.strftime("%m-%d"))

        # 查询时间A到B这一天的注册人数
        everyday_active_count = User.query.filter(User.is_admin == False, User.last_login >= begin_date,
                                                  User.last_login <= end_date).count()

        # 添加当天注册人数到,获取数量中
        active_count.append(everyday_active_count)

    #为了方便查看图标,反转
    active_date.reverse()
    active_count.reverse()

    # - 5.携带数据,渲染页面
    data = {
        "total_count":total_count,
        "month_count":month_count,
        "day_count":day_count,
        "active_date":active_date,
        "active_count":active_count
    }
    return render_template("admin/user_count.html",data=data)



# 功能描述: 管理首页
# 请求路径: /admin/index
# 请求方式: GET
# 请求参数: 无
# 返回值:渲染页面index.html,user字典数据
@admin_blue.route('/index')
@user_login_data
def admin_index():

    admin = g.user.to_dict() if g.user else ""
    return render_template("admin/index.html",admin=admin)


#功能描述:管理员登陆
# 请求路径: /admin/login
# 请求方式: GET,POST
# 请求参数:GET,无, POST,username,password
# 返回值: GET渲染login.html页面, POST,login.html页面,errmsg
@admin_blue.route('/login',methods=["GET","POST"])
def admin_login():
    """
    - 1.判断请求方式,是否GET,直接渲染页面
    - 2.如果是POST,获取参数
    - 3.根据用户名,查询管理员对象
    - 4.判断管理员对象是否存在
    - 5.校验管理员密码是否正确
    - 6.记录管理员登陆信息到session
    - 7.重定向到首页
    :return:
    """
    # - 1.判断请求方式,是否GET,直接渲染页面
    if request.method == "GET":

        #判断管理员是否已经登陆过,如果登陆过,直接重定向到首页
        if session.get("is_admin"):
            return redirect("/admin/index")

        return render_template("admin/login.html")

    # - 2.如果是POST,获取参数
    username = request.form.get("username")
    password = request.form.get("password")

    # - 3.根据用户名,查询管理员对象
    try:
        admin = User.query.filter(User.mobile == username,User.is_admin == True).first()
    except Exception as e:
        current_app.logger.error(e)
        return render_template("admin/login.html",errmsg="管理员登陆失败")

    # - 4.判断管理员对象是否存在
    if not admin:
        return render_template("admin/login.html",errmsg="管理员不存在")

    # - 5.校验管理员密码是否正确
    if not admin.check_passowrd(password):
        return render_template("admin/login.html",errmsg="密码错误")

    # - 6.记录管理员登陆信息到session
    session["user_id"] = admin.id
    session["nick_name"] = admin.nick_name
    session["mobile"] = admin.mobile
    session["is_admin"] = True

    # - 7.重定向到首页(未完成)
    return redirect("/admin/index")

